package com.dkd.manage.service.impl;

import java.time.Duration;
import java.util.List;
import java.util.stream.Collectors;

import cn.hutool.core.bean.BeanUtil;
import cn.hutool.core.collection.CollUtil;
import cn.hutool.core.util.StrUtil;
import com.dkd.common.constant.DkdContants;
import com.dkd.common.exception.ServiceException;
import com.dkd.common.utils.DateUtils;
import com.dkd.manage.domain.Emp;
import com.dkd.manage.domain.TaskDetails;
import com.dkd.manage.domain.VendingMachine;
import com.dkd.manage.domain.dto.TaskDetailsDto;
import com.dkd.manage.domain.dto.TaskDto;
import com.dkd.manage.domain.vo.TaskVo;
import com.dkd.manage.service.IEmpService;
import com.dkd.manage.service.ITaskDetailsService;
import com.dkd.manage.service.IVendingMachineService;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.data.redis.core.RedisTemplate;
import org.springframework.stereotype.Service;
import com.dkd.manage.mapper.TaskMapper;
import com.dkd.manage.domain.Task;
import com.dkd.manage.service.ITaskService;
import org.springframework.transaction.annotation.Transactional;

/**
 * 工单Service业务层处理
 * 
 * @author jrz
 * @date 2025-09-10
 */
@Service
public class TaskServiceImpl implements ITaskService 
{
    @Autowired
    private TaskMapper taskMapper;

    /**
     * 查询工单
     * 
     * @param taskId 工单主键
     * @return 工单
     */
    @Override
    public Task selectTaskByTaskId(Long taskId)
    {
        return taskMapper.selectTaskByTaskId(taskId);
    }

    /**
     * 查询工单列表
     * 
     * @param task 工单
     * @return 工单
     */
    @Override
    public List<Task> selectTaskList(Task task)
    {
        return taskMapper.selectTaskList(task);
    }

    /**
     * 新增工单
     * 
     * @param task 工单
     * @return 结果
     */
    @Override
    public int insertTask(Task task)
    {
        task.setCreateTime(DateUtils.getNowDate());
        return taskMapper.insertTask(task);
    }

    /**
     * 修改工单
     * 
     * @param task 工单
     * @return 结果
     */
    @Override
    public int updateTask(Task task)
    {
        task.setUpdateTime(DateUtils.getNowDate());
        return taskMapper.updateTask(task);
    }

    /**
     * 批量删除工单
     * 
     * @param taskIds 需要删除的工单主键
     * @return 结果
     */
    @Override
    public int deleteTaskByTaskIds(Long[] taskIds)
    {
        return taskMapper.deleteTaskByTaskIds(taskIds);
    }

    /**
     * 删除工单信息
     * 
     * @param taskId 工单主键
     * @return 结果
     */
    @Override
    public int deleteTaskByTaskId(Long taskId)
    {
        return taskMapper.deleteTaskByTaskId(taskId);
    }
    /**
     * 查询工单列表
     * @param task
     * @return TaskVo集合
     */
    @Override
    public List<TaskVo> selectTaskVoList(Task task) {
        return taskMapper.selectTaskVoList(task);
    }

    @Autowired
    private IVendingMachineService vendingMachineService;
    @Autowired
    private IEmpService empService;
    @Autowired
    private RedisTemplate redisTemplate;
    @Autowired
    private ITaskDetailsService  taskDetailsService;
    /**
     * 新增运营，运维工单
     * @param taskDto
     * @return
     */
    @Transactional
    @Override
    public int insertTaskDto(TaskDto taskDto) {
        //1.查询售货机是否存在
        VendingMachine vm = vendingMachineService.selectVendingMachineByInnerCode(taskDto.getInnerCode());
        if(vm==null){
            throw new ServiceException("设备不存在");
        }
        //2.校验售货机状态与工单类型是否相符
        checkCreateTask(vm.getVmStatus(),taskDto.getProductTypeId());
        //3.检查设备是否有未完成的同类型工单
        hasTask(taskDto);
        //4.查询并校验员工是否存在
        Emp emp = empService.selectEmpById(taskDto.getUserId());
        if(emp==null){
            throw new ServiceException("员工不存在");
        }
        //5.校验员工区域是否匹配
        if(!emp.getRegionId().equals(vm.getRegionId())){
            throw new ServiceException("员工区域与设备区域不匹配，无法处理此工单");
        }
        //6.将dto转为po并补充属性，保存工单
        Task task = BeanUtil.copyProperties(taskDto, Task.class);
        task.setTaskStatus(DkdContants.TASK_STATUS_CREATE);//创建工单
        task.setUserName(emp.getUserName());//执行人名称
        task.setRegionId(vm.getRegionId());//所属区域id
        task.setAddr(vm.getAddr());//地址
        task.setCreateTime(DateUtils.getNowDate());//创建时间
        task.setTaskCode(generateTaskCode());//工单编号
        int taskResult = taskMapper.insertTask(task);
        //7.判断是否为补货工单
        if(taskDto.getProductTypeId().equals(DkdContants.TASK_TYPE_SUPPLY)){
            //8.保存工单详情
            List<TaskDetailsDto> details=taskDto.getDetails();
            if (CollUtil.isEmpty(details)) {
                throw new ServiceException("补货工单详情不能为空");
            }
            //将dto转为po补充属性
            List<TaskDetails> taskDetailsList = details.stream().map(detailsDto -> {
                TaskDetails taskDetails = BeanUtil.copyProperties(detailsDto, TaskDetails.class);
                taskDetails.setTaskId(task.getTaskId());
                return taskDetails;
            }).collect(Collectors.toList());
            //批量新增
            taskDetailsService.batchInsertTaskDetails(taskDetailsList);
        }

        return taskResult;
    }
    // 生成并获取当天的工单编号（唯一标识）
    private String generateTaskCode(){
        // 获取当前日期并格式化为"yyyyMMdd"
        String dateStr = DateUtils.getDate().replaceAll("-","");
        // 根据日期来生成redis的键
        String key = "dkd.task.code." + dateStr;
        try {
            // 使用Redis原子操作生成序列号
            // 直接执行自增操作，Redis会自动处理key不存在的情况
            Long incrementValue = redisTemplate.opsForValue().increment(key, 1);
            // 如果是第一次创建（值为1），设置过期时间
            if (incrementValue == 1) {
                redisTemplate.expire(key, Duration.ofDays(1));
            }
            // 格式化工单编号，确保为4位数字
            return dateStr + StrUtil.padPre(incrementValue.toString(), 4, "0");
        } catch (Exception e) {
            // 如果发生异常，尝试删除key并重新设置
            try {
                redisTemplate.delete(key);
                redisTemplate.opsForValue().set(key, "1", Duration.ofDays(1));
                return dateStr + "0001";
            } catch (Exception ex) {
                // 如果仍然失败，抛出异常
                throw new ServiceException("生成工单编号失败: " + ex.getMessage());
            }
        }
    }
    //检查设备是否有未完成的同类型工单
    private void hasTask(TaskDto taskDto) {
        //创建task条件对象，并设置设备编号和工单类型，以及工单状态为进行中
        Task taskParam = new Task();
        taskParam.setInnerCode(taskDto.getInnerCode());
        taskParam.setProductTypeId(taskDto.getProductTypeId());
        taskParam.setTaskStatus(DkdContants.TASK_STATUS_PROGRESS);
        //调用taskMapper查询数据库是否有符合条件的工单列表：
        List<Task> taskList = taskMapper.selectTaskList(taskParam);
        //如果存在未完成的同类型工单，抛出异常
        if(taskList!=null&&taskList.size()>0){
            throw new ServiceException("该设备有未完成的工单，不能重复创建");
        }
    }

    private void checkCreateTask(Long vmStatus,Long productTypeId){
        //如果是投放工单，设备在运行中，抛出异常
        if (productTypeId== DkdContants.TASK_TYPE_DEPLOY&&vmStatus== DkdContants.VM_STATUS_RUNNING){
            throw new ServiceException("该设备状态为运行中，无法进行投放");
        }
        //如果是维修工单，设备不在运行中，抛出异常
        if(productTypeId== DkdContants.TASK_TYPE_REPAIR&&vmStatus!= DkdContants.VM_STATUS_RUNNING){
            throw new ServiceException("该设备状态非运行中，无法进行维修");
        }
        //如果是补货工单，设备不在运行中，抛出异常
        if(productTypeId== DkdContants.TASK_TYPE_SUPPLY&&vmStatus!= DkdContants.VM_STATUS_RUNNING){
            throw new ServiceException("该设备状态非运行中，无法进行补货");
        }
        //如果是撤机工单，设备不在运行中，抛出异常
        if(productTypeId== DkdContants.TASK_TYPE_REVOKE&&vmStatus!= DkdContants.VM_STATUS_RUNNING){
            throw new ServiceException("该设备状态非运行中，无法进行撤机");
        }
    }


    /**
     * 取消工单
     * @param task
     * @return
     */
    @Override
    public int cancelTask(Task task) {
        //1.判断工单状态是否可以取消
        //先根据工单id查询数据库
        Task taskDb = taskMapper.selectTaskByTaskId(task.getTaskId());
        //判断工单状态是否为已取消
        if(taskDb.getTaskStatus().equals(DkdContants.TASK_STATUS_CANCEL)){
            throw new ServiceException("该工单已取消，不能再次取消");
        }
        //判断工单状态是否为已完成，如果是，则抛出异常
        if(taskDb.getTaskStatus().equals(DkdContants.TASK_STATUS_FINISH)){
            throw new ServiceException("该工单已完成，不能取消");
        }
        //2.设置更新字段
        task.setTaskStatus(DkdContants.TASK_STATUS_CANCEL);//工单状态：取消
        task.setUpdateTime(DateUtils.getNowDate());//更新时间
        //3.更新工单
        return taskMapper.updateTask(task);//注意别传错了，这里是前端的task参数，而不是数据库的taskDb参数
    }
}
